Skill

লিংকড লিস্ট (Linked Lists)

Computer Science - ডাটা স্ট্রাকচার & অ্যালগরিদম (Data Structure & Algorithms)
414

লিঙ্কড লিস্ট (Linked List) একটি ডেটা স্ট্রাকচার যা ডাইনামিক্যালি ডেটা সংরক্ষণ করতে ব্যবহৃত হয়। এটি নোড (Node) এর একটি শ্রেণী, যেখানে প্রতিটি নোডে একটি ডেটা অংশ এবং পরবর্তী নোডের রেফারেন্স (পয়েন্টার) থাকে। লিঙ্কড লিস্টগুলি সাধারণত অ্যারে থেকে বেশি স্থিতিশীল এবং সহজে সম্প্রসারণযোগ্য।

লিঙ্কড লিস্টের প্রধান বৈশিষ্ট্য

  1. ডাইনামিক মেমরি: লিঙ্কড লিস্টের আকার চলাকালীন সময়ে পরিবর্তিত হতে পারে, যা মেমরি ব্যবস্থাপনার ক্ষেত্রে সুবিধা প্রদান করে।
  2. সহজ ইনসার্ট এবং ডিলিট: নতুন উপাদান যুক্ত করা বা বিদ্যমান উপাদান মুছে ফেলা সহজ, বিশেষ করে শুরু এবং শেষে।
  3. সর্বদা sequential access: এলিমেন্টগুলি অ্যাক্সেস করতে হলে লিঙ্কড লিস্টে প্রতিটি নোড ধরে ধরে যেতে হয়।

লিঙ্কড লিস্টের প্রকারভেদ

১. সিঙ্গল লিঙ্কড লিস্ট (Singly Linked List):

  • প্রতিটি নোডের মধ্যে একটি ডেটা অংশ এবং পরবর্তী নোডের জন্য একটি পয়েন্টার থাকে।
  • সর্বশেষ নোডের পরবর্তী পয়েন্টার NULL নির্দেশ করে।

২. ডাবল লিঙ্কড লিস্ট (Doubly Linked List):

  • প্রতিটি নোডে একটি ডেটা অংশ, পূর্ববর্তী নোডের পয়েন্টার এবং পরবর্তী নোডের পয়েন্টার থাকে।
  • এটি উভয় দিকে চলাফেরা করতে পারে।

৩. সার্কুলার লিঙ্কড লিস্ট (Circular Linked List):

  • সিঙ্গল বা ডাবল লিঙ্কড লিস্ট হতে পারে, তবে সর্বশেষ নোডটি প্রথম নোডের সাথে সংযুক্ত থাকে, ফলে এটি একটি লুপ তৈরি করে।

লিঙ্কড লিস্টের কার্যকারিতা

১. নতুন নোড তৈরি করা:

  • নতুন নোড তৈরি করতে এবং তাকে যুক্ত করতে হবে।

২. নোড ইনসার্ট করা:

  • শুরু, মাঝের, বা শেষের অংশে নতুন নোড যুক্ত করা।

৩. নোড মুছে ফেলা:

  • নির্দিষ্ট মান বা ইনডেক্সের ভিত্তিতে নোড মুছে ফেলা।

৪. তথ্য অ্যাক্সেস:

  • লিঙ্কড লিস্টে তথ্য অ্যাক্সেস করতে হলে প্রতিটি নোড ধরে ধরে যেতে হয়।

উদাহরণ (পাইথনে সিঙ্গল লিঙ্কড লিস্ট)

class Node:
    def __init__(self, data):
        self.data = data
        self.next = None

class LinkedList:
    def __init__(self):
        self.head = None

    def append(self, data):
        new_node = Node(data)
        if not self.head:
            self.head = new_node
            return
        last_node = self.head
        while last_node.next:
            last_node = last_node.next
        last_node.next = new_node

    def display(self):
        current_node = self.head
        while current_node:
            print(current_node.data, end=" -> ")
            current_node = current_node.next
        print("None")

# ব্যবহার
llist = LinkedList()
llist.append(1)
llist.append(2)
llist.append(3)
llist.display()  # আউটপুট: 1 -> 2 -> 3 -> None

উপসংহার

লিঙ্কড লিস্ট একটি শক্তিশালী ডেটা স্ট্রাকচার যা ডেটা সংরক্ষণ ও পরিচালনার জন্য নমনীয়তা প্রদান করে। এর ডাইনামিক প্রকৃতি এবং সহজ ইনসার্ট এবং ডিলিট অপারেশন লিঙ্কড লিস্টকে বিশেষ করে বড় ডেটাসেটের সাথে কাজ করার জন্য উপযুক্ত করে তোলে। এটি বিভিন্ন ধরনের ডেটা স্ট্রাকচারের মধ্যে একটি মৌলিক ভিত্তি।

লিঙ্কড লিস্টের ধারণা এবং প্রকারভেদ: Singly, Doubly, Circular Linked List

927

লিঙ্কড লিস্টের ধারণা

লিঙ্কড লিস্ট হলো একটি ডেটা স্ট্রাকচার যেখানে ডেটা উপাদানগুলো একটি ধারাবাহিক নোডে সংরক্ষিত থাকে, এবং প্রতিটি নোড তার পরবর্তী নোডের সাথে যুক্ত থাকে। লিঙ্কড লিস্টের প্রতিটি নোডে দুটি প্রধান অংশ থাকে:

  1. ডেটা অংশ: এখানে উপাদানের মান সংরক্ষিত থাকে।
  2. পরবর্তী অংশ (Next/Pointer): এটি পরবর্তী নোডের ঠিকানা ধারণ করে।

লিঙ্কড লিস্টের প্রধান সুবিধা হলো এটি ডায়নামিক মেমরি ব্যবস্থাপনা সমর্থন করে, যার মাধ্যমে সহজে নতুন উপাদান যোগ বা মুছে ফেলা যায়। অ্যারের মতো পূর্বনির্ধারিত আকারের প্রয়োজন হয় না।


লিঙ্কড লিস্টের প্রকারভেদ

লিঙ্কড লিস্ট সাধারণত তিন প্রকারের হয়ে থাকে:

  1. সিঙ্গলি লিঙ্কড লিস্ট (Singly Linked List)
  2. ডাবলি লিঙ্কড লিস্ট (Doubly Linked List)
  3. সার্কুলার লিঙ্কড লিস্ট (Circular Linked List)

১. সিঙ্গলি লিঙ্কড লিস্ট (Singly Linked List)

সিঙ্গলি লিঙ্কড লিস্ট হলো সবচেয়ে সহজ লিঙ্কড লিস্ট প্রকার যেখানে প্রতিটি নোড শুধুমাত্র পরবর্তী নোডের ঠিকানা ধারণ করে। এটি শুধুমাত্র একদিকে চলতে পারে।

বৈশিষ্ট্য:

  • প্রতিটি নোডে একটি ডেটা অংশ এবং একটি পরবর্তী নোডের পয়েন্টার থাকে।
  • প্রথম নোডকে হেড (Head) বলা হয়, এবং এটি লিঙ্কড লিস্টের সূচনা।
  • শেষ নোডের পয়েন্টার NULL হয়, কারণ এর পর কোন নোড নেই।

উদাহরণ (C++):

#include <iostream>
using namespace std;

struct Node {
    int data;
    Node* next;
};

int main() {
    Node* head = new Node(); // হেড নোড তৈরি করা
    head->data = 1;
    head->next = new Node(); // দ্বিতীয় নোড তৈরি করা
    head->next->data = 2;
    head->next->next = NULL; // শেষ নোড, তাই পরবর্তী NULL

    // লিস্ট প্রদর্শন করা
    Node* current = head;
    while (current != NULL) {
        cout << current->data << " ";
        current = current->next;
    }

    return 0;
}

২. ডাবলি লিঙ্কড লিস্ট (Doubly Linked List)

ডাবলি লিঙ্কড লিস্ট হলো এমন একটি লিঙ্কড লিস্ট যেখানে প্রতিটি নোডে একটি ডেটা অংশ, একটি পরবর্তী নোডের পয়েন্টার এবং একটি পূর্ববর্তী নোডের পয়েন্টার থাকে। এটি উভয় দিকে চলতে পারে।

বৈশিষ্ট্য:

  • প্রতিটি নোডে একটি Prev এবং একটি Next পয়েন্টার থাকে।
  • প্রথম নোডের পূর্ববর্তী পয়েন্টার NULL হয় এবং শেষ নোডের পরবর্তী পয়েন্টার NULL হয়।
  • উভয় দিকে লিস্ট ট্রাভার্স করা যায়।

উদাহরণ (C++):

#include <iostream>
using namespace std;

struct Node {
    int data;
    Node* next;
    Node* prev;
};

int main() {
    Node* head = new Node();
    head->data = 1;
    head->next = new Node();
    head->next->prev = head;
    head->next->data = 2;
    head->next->next = NULL;

    // ডাবলি লিঙ্কড লিস্ট প্রদর্শন করা
    Node* current = head;
    while (current != NULL) {
        cout << current->data << " ";
        current = current->next;
    }

    return 0;
}

৩. সার্কুলার লিঙ্কড লিস্ট (Circular Linked List)

সার্কুলার লিঙ্কড লিস্ট এমন একটি লিঙ্কড লিস্ট যেখানে শেষ নোডের পয়েন্টার প্রথম নোডের দিকে নির্দেশ করে। এটি একটি চক্র তৈরি করে, যাতে কোনও নির্দিষ্ট শেষ নেই।

বৈশিষ্ট্য:

  • সিঙ্গলি এবং ডাবলি লিঙ্কড লিস্ট উভয়ের মতোই হতে পারে।
  • শেষ নোডের পরবর্তী পয়েন্টার প্রথম নোডকে নির্দেশ করে, ফলে পুরো লিস্ট ঘুরতে থাকে।

উদাহরণ (C++):

#include <iostream>
using namespace std;

struct Node {
    int data;
    Node* next;
};

int main() {
    Node* head = new Node();
    head->data = 1;
    head->next = new Node();
    head->next->data = 2;
    head->next->next = head;  // শেষ নোডটি প্রথম নোডকে নির্দেশ করে

    // সার্কুলার লিঙ্কড লিস্ট প্রদর্শন করা (সতর্কতাঃ সীমিত প্রদর্শন)
    Node* current = head;
    int count = 0;
    do {
        cout << current->data << " ";
        current = current->next;
        count++;
    } while (current != head && count < 5);

    return 0;
}

লিঙ্কড লিস্টের তুলনামূলক বৈশিষ্ট্য

প্রকারদিকনির্দেশমেমরি খরচসুবিধাঅসুবিধা
Singly Linked Listএকদিককমসহজ এবং কম মেমরি খরচউল্টো দিকে ট্রাভার্স করা যায় না
Doubly Linked Listদুইদিকবেশিউভয় দিকে ট্রাভার্স করা যায়অতিরিক্ত মেমরি খরচ
Circular Linked Listএক বা দুইদিককম/বেশিঅবিরাম ট্রাভার্স করা যায়স্টপ কন্ডিশন ছাড়া ইনফিনিট লুপ হতে পারে

উপসংহার

লিঙ্কড লিস্ট ডেটা সংরক্ষণ এবং পরিচালনার একটি কার্যকর ডেটা স্ট্রাকচার, যা ডায়নামিক মেমরি ব্যবস্থাপনা এবং ডেটা অ্যাক্সেসকে সহজ করে। সিঙ্গলি, ডাবলি এবং সার্কুলার লিঙ্কড লিস্টের প্রতিটি প্রকার বিভিন্ন পরিস্থিতিতে কার্যকর ভূমিকা পালন করে, এবং তাদের বৈশিষ্ট্য অনুসারে বিভিন্ন কাজে ব্যবহার করা হয়।

লিঙ্কড লিস্টে ইনসার্ট, ডিলিট এবং সার্চ অপারেশন

191

লিঙ্কড লিস্ট হল একটি ডেটা স্ট্রাকচার যা উপাদানগুলি (নোড) লিঙ্কের মাধ্যমে সংযুক্ত থাকে। প্রতিটি নোডে একটি ডেটা এবং পরবর্তী নোডের জন্য একটি রেফারেন্স (পয়েন্টার) থাকে। লিঙ্কড লিস্টের কিছু মৌলিক অপারেশন হল ইনসার্ট, ডিলিট এবং সার্চ। নিচে এই অপারেশনগুলোর বিস্তারিত আলোচনা করা হলো।

লিঙ্কড লিস্টের গঠন

প্রথমে, একটি লিঙ্কড লিস্টের নোডের গঠন এবং লিঙ্কড লিস্ট তৈরি করা যাক।

class Node:
    def __init__(self, data):
        self.data = data
        self.next = None  # পরবর্তী নোডের জন্য পয়েন্টার

class LinkedList:
    def __init__(self):
        self.head = None  # লিঙ্কড লিস্টের মাথা

১. ইনসার্ট অপারেশন (Insert Operation)

নতুন নোড যুক্ত করার জন্য লিঙ্কড লিস্টে বিভিন্ন উপায়ে ইনসার্ট করা যেতে পারে:

১.১ মাথায় ইনসার্ট (Insert at Head)

def insert_at_head(self, data):
    new_node = Node(data)  # নতুন নোড তৈরি
    new_node.next = self.head  # নতুন নোডের পরবর্তী নোড পুরানো মাথা
    self.head = new_node  # মাথা নতুন নোডকে নির্দেশ করে

১.২ শেষে ইনসার্ট (Insert at Tail)

def insert_at_tail(self, data):
    new_node = Node(data)
    if not self.head:
        self.head = new_node  # যদি মাথা খালি হয়, নতুন নোড মাথা হয়
        return
    last_node = self.head
    while last_node.next:
        last_node = last_node.next  # শেষ নোডে পৌঁছান
    last_node.next = new_node  # নতুন নোড যুক্ত করুন

২. ডিলিট অপারেশন (Delete Operation)

লিঙ্কড লিস্ট থেকে একটি নির্দিষ্ট নোড মুছে ফেলার জন্য ডিলিট অপারেশন ব্যবহার করা হয়।

২.১ নির্দিষ্ট নোড ডিলিট (Delete Specific Node)

def delete_node(self, key):
    current_node = self.head

    # যদি মাথা নিজেই মুছে ফেলতে হয়
    if current_node and current_node.data == key:
        self.head = current_node.next  # মাথাকে আপডেট করুন
        current_node = None  # নোড মুছুন
        return

    # অন্য নোড মুছে ফেলুন
    previous_node = None
    while current_node and current_node.data != key:
        previous_node = current_node
        current_node = current_node.next

    # যদি নোড না পাওয়া যায়
    if not current_node:
        return

    previous_node.next = current_node.next  # নোডটি বাদ দিন
    current_node = None  # নোড মুছুন

৩. সার্চ অপারেশন (Search Operation)

লিঙ্কড লিস্টে একটি নির্দিষ্ট মান খুঁজে বের করার জন্য সার্চ অপারেশন ব্যবহার করা হয়।

সার্চ অপারেশন

def search(self, key):
    current_node = self.head
    while current_node:
        if current_node.data == key:
            return True  # পাওয়া গেছে
        current_node = current_node.next
    return False  # পাওয়া যায়নি

লিঙ্কড লিস্টের উদাহরণ

এখন একটি সম্পূর্ণ উদাহরণ দিয়ে দেখানো যাক, যেখানে ইনসার্ট, ডিলিট এবং সার্চ অপারেশন একত্রে ব্যবহার করা হয়েছে।

class Node:
    def __init__(self, data):
        self.data = data
        self.next = None

class LinkedList:
    def __init__(self):
        self.head = None

    def insert_at_head(self, data):
        new_node = Node(data)
        new_node.next = self.head
        self.head = new_node

    def insert_at_tail(self, data):
        new_node = Node(data)
        if not self.head:
            self.head = new_node
            return
        last_node = self.head
        while last_node.next:
            last_node = last_node.next
        last_node.next = new_node

    def delete_node(self, key):
        current_node = self.head
        if current_node and current_node.data == key:
            self.head = current_node.next
            current_node = None
            return
        previous_node = None
        while current_node and current_node.data != key:
            previous_node = current_node
            current_node = current_node.next
        if not current_node:
            return
        previous_node.next = current_node.next
        current_node = None

    def search(self, key):
        current_node = self.head
        while current_node:
            if current_node.data == key:
                return True
            current_node = current_node.next
        return False

    def print_list(self):
        current_node = self.head
        while current_node:
            print(current_node.data, end=" -> ")
            current_node = current_node.next
        print("None")

# উদাহরণ ব্যবহারের জন্য
linked_list = LinkedList()
linked_list.insert_at_head(10)
linked_list.insert_at_tail(20)
linked_list.insert_at_tail(30)

print("Linked List:")
linked_list.print_list()  # 10 -> 20 -> 30 -> None

linked_list.delete_node(20)
print("After deleting 20:")
linked_list.print_list()  # 10 -> 30 -> None

print("Search for 30:", linked_list.search(30))  # True
print("Search for 20:", linked_list.search(20))  # False

উপসংহার

লিঙ্কড লিস্টে ইনসার্ট, ডিলিট এবং সার্চ অপারেশনগুলি একটি গুরুত্বপূর্ণ ডেটা স্ট্রাকচারের কার্যকারিতা প্রদর্শন করে। এগুলো বিভিন্ন পরিস্থিতিতে ডেটা সংরক্ষণ এবং পরিচালনার জন্য ব্যবহৃত হয়। লিঙ্কড লিস্টের এই মৌলিক অপারেশনগুলোকে ব্যবহার করে আপনি ডেটার প্রয়োজনীয়তা অনুযায়ী কোড তৈরি করতে পারেন।

লিঙ্কড লিস্টের অ্যাপ্লিকেশন এবং জটিলতা বিশ্লেষণ

155


লিঙ্কড লিস্ট (Linked List)

লিঙ্কড লিস্ট হল একটি ডেটা স্ট্রাকচার যেখানে ডেটার উপাদানগুলি (নোড) একটি সিকোয়েন্সে সংযুক্ত থাকে। প্রতিটি নোডে একটি ডেটা ফিল্ড এবং পরবর্তী নোডের জন্য একটি রেফারেন্স (পয়েন্টার) থাকে। এটি অ্যারে থেকে ভিন্ন কারণ এটি ডাইনামিকভাবে মেমরি বরাদ্দ করতে সক্ষম।

লিঙ্কড লিস্টের অ্যাপ্লিকেশন

লিঙ্কড লিস্টের অনেকগুলি প্রয়োগ রয়েছে, কিছু গুরুত্বপূর্ণ অ্যাপ্লিকেশন হলো:

১. ডায়নামিক মেমরি ব্যবস্থাপনা:

  • ডাইনামিকভাবে ডেটা সংরক্ষণ এবং মুছে ফেলা যায়। অ্যারের তুলনায় এটি বড় আকারের ডেটা হ্যান্ডলিং সহজ করে।

২. স্ট্যাক এবং কিউ:

  • স্ট্যাক (Last In First Out) এবং কিউ (First In First Out) ডেটা স্ট্রাকচার হিসাবে লিঙ্কড লিস্ট ব্যবহার করা হয়। উদাহরণস্বরূপ, ফাংশন কলের ইতিহাস সংরক্ষণ করতে স্ট্যাক।

৩. অপারেটিং সিস্টেমে প্রক্রিয়া ব্যবস্থাপনা:

  • লিঙ্কড লিস্ট প্রক্রিয়া নিয়ন্ত্রণ ব্লক (PCB) এবং রানকিউ পরিচালনার জন্য ব্যবহৃত হয়।

৪. অ্যালগরিদমে ব্যবহার:

  • বিভিন্ন অ্যালগরিদম যেমন সোর্টিং এবং মার্জিংয়ে ডেটার অর্গানাইজেশন করার জন্য।

৫. ডেটাবেসের সংযোগ:

  • সম্পর্কিত রেকর্ডের মধ্যে লিঙ্ক তৈরি করতে ব্যবহৃত হয়, যা ফাইল এবং ডাটাবেসের মধ্যে যোগাযোগ করে।

৬. ইমেজ প্রক্রিয়াকরণ:

  • ইমেজ পিক্সেলের জন্য লিঙ্কড লিস্ট ব্যবহার করা হয়, যেখানে প্রতিটি পিক্সেল একে অপরের সাথে যুক্ত থাকে।

জটিলতা বিশ্লেষণ

লিঙ্কড লিস্টের অপারেশনগুলোর জন্য জটিলতা বিশ্লেষণ করা হয়েছে:

১. ইনসার্ট করা:

  • অগ্রভাগে (Head): O(1)
  • পেছনে (Tail): O(n) (যদি শেষ নোডে পৌঁছানোর জন্য পুরো লিঙ্কড লিস্ট পাস করতে হয়)
  • মধ্যবর্তী: O(n) (নোড খুঁজে পাওয়ার জন্য)

২. ডিলিট করা:

  • অগ্রভাগ থেকে: O(1)
  • পেছনে: O(n)
  • মধ্যবর্তী: O(n) (নোড খুঁজে পাওয়ার জন্য)

৩. সার্চ করা:

  • O(n) (নোড খুঁজে পাওয়ার জন্য লিঙ্কড লিস্টের সবগুলো নোড পাস করতে হয়)

৪. কন্ট্রোল ট্রাভার্সাল:

  • O(n) (লিঙ্কড লিস্টের সবগুলো নোড পাস করতে হয়)

উপসংহার

লিঙ্কড লিস্ট হল একটি শক্তিশালী এবং কার্যকরী ডেটা স্ট্রাকচার যা বিভিন্ন অ্যাপ্লিকেশন এবং সমস্যার সমাধানে ব্যবহার করা হয়। এর ডাইনামিক মেমরি ব্যবস্থাপনা এবং ইনসার্ট/ডিলিট অপারেশনে সুবিধা রয়েছে, তবে সার্চ অপারেশনে অ্যারের তুলনায় কিছুটা ধীর। এটি প্রোগ্রামিং এবং ডেটা স্ট্রাকচার ডিজাইনে একটি গুরুত্বপূর্ণ অংশ।

Promotion
NEW SATT AI এখন আপনাকে সাহায্য করতে পারে।

Are you sure to start over?

Loading...